import { products, referralPrograms } from "../../global.js";
import { assert } from "chai";
import chalk from "chalk";
import AutumnError, { AutumnInt } from "@/external/autumn/autumnCli.js";
import { setupBefore } from "tests/before.js";
import {
  Customer,
  ErrCode,
  ReferralCode,
  RewardRedemption,
} from "@autumn/shared";
import { timeout } from "tests/utils/genUtils.js";
import { initCustomerWithTestClock } from "tests/utils/testInitUtils.js";
import { advanceTestClock } from "tests/utils/stripeUtils.js";
import { addDays, addHours, addMonths } from "date-fns";
import { Stripe } from "stripe";
import { initCustomer } from "tests/utils/init.js";

// UNCOMMENT FROM HERE
describe(`${chalk.yellowBright(
  "referrals2: Testing referrals (immediate redemption)",
)}`, () => {
  let mainCustomerId = "main-referral-2";
  let redeemers = ["referral2-r1", "referral2-r2", "referral2-r3"];
  let autumn: AutumnInt = new AutumnInt();
  let stripeCli: Stripe;
  let testClockId: string;
  let referralCode: ReferralCode;

  let redemptions: RewardRedemption[] = [];
  let mainCustomer: Customer;

  before(async function () {
    await setupBefore(this);
    stripeCli = this.stripeCli;

    const { testClockId: testClockId1, customer } =
      await initCustomerWithTestClock({
        customerId: mainCustomerId,
        db: this.db,
        org: this.org,
        env: this.env,
      });
    testClockId = testClockId1;
    mainCustomer = customer;

    let batchCreate = [];
    for (let redeemer of redeemers) {
      batchCreate.push(
        initCustomer({
          customerId: redeemer,
          db: this.db,
          org: this.org,
          env: this.env,
          attachPm: true,
        }),
      );
    }

    await Promise.all(batchCreate);
  });

  it("should create code once", async function () {
    referralCode = await autumn.referrals.createCode({
      customerId: mainCustomerId,
      referralId: referralPrograms.immediate.id,
    });

    assert.exists(referralCode.code);
  });

  it("should create redemption for each redeemer and fail if redeemed again", async function () {
    for (let i = 0; i < redeemers.length; i++) {
      let redeemer = redeemers[i];
      let count = i + 1;
      try {
        let redemption: RewardRedemption = await autumn.referrals.redeem({
          customerId: redeemer,
          code: referralCode.code,
        });
        redemptions.push(redemption);

        if (count > referralPrograms.immediate.max_redemptions) {
          assert.equal(redemption.triggered, false);
          assert.equal(redemption.applied, false);
        } else {
          assert.fail("Should not be able to redeem again");
        }
      } catch (error) {
        if (count > referralPrograms.immediate.max_redemptions) {
          assert.instanceOf(error, AutumnError);
          assert.equal(error.code, ErrCode.ReferralCodeMaxRedemptionsReached);
        }
      }
    }

    // Try redeem for redeemer1 again
    // try {
    //   let redemption1 = await autumn.referrals.redeem({
    //     customerId: redeemers[0],
    //     code: referralCode.code,
    //   });
    //   assert.fail("Should not be able to redeem again");
    // } catch (error) {
    //   assert.instanceOf(error, AutumnError);
    //   assert.equal(error.code, ErrCode.CustomerAlreadyRedeemedReferralCode);
    // }

    // Check stripe customer
    let stripeCus = (await stripeCli.customers.retrieve(
      mainCustomer.processor?.id,
    )) as Stripe.Customer;

    assert.notEqual(stripeCus.discount, null);
  });

  let curTime = new Date();
  it("customer should have discount for first purchase", async function () {
    await autumn.attach({
      customer_id: mainCustomerId,
      product_id: products.proWithTrial.id,
    });

    await timeout(3000);

    curTime = addDays(addDays(curTime, 7), 1);
    await advanceTestClock({
      testClockId,
      advanceTo: curTime.getTime(),
      stripeCli,
    });

    // 1. Get invoice
    let { invoices } = await autumn.customers.get(mainCustomerId);

    assert.equal(invoices!.length, 2);
    assert.equal(invoices![0].total, 0);
  });

  // it("customer should have discount for second purchase", async function () {
  //   // 2. Check that customer has another discount
  //   let stripeCus = (await stripeCli.customers.retrieve(
  //     mainCustomer.processor?.id,
  //   )) as Stripe.Customer;

  //   assert.notEqual(stripeCus.discount, null);

  //   // 2. Advance test clock to 1 month from start (trigger discount.deleted event)
  //   curTime = addHours(addMonths(new Date(), 1), 2);
  //   await advanceTestClock({
  //     testClockId,
  //     advanceTo: curTime.getTime(),
  //     stripeCli,
  //   });

  //   // 3. Advance test clock to 1 month + 7 days from start (trigger new invoice)
  //   curTime = addDays(curTime, 8);
  //   await advanceTestClock({
  //     testClockId,
  //     advanceTo: curTime.getTime(),
  //     stripeCli,
  //   });

  //   // // 3. Get invoice again
  //   let { invoices: invoices2 } = await autumn.customers.get(mainCustomerId);

  //   assert.equal(invoices2!.length, 3);
  //   assert.equal(invoices2![0].total, 0);
  // });
});
